home *** CD-ROM | disk | FTP | other *** search
/ CU Amiga Super CD-ROM 15 / CU Amiga Magazine's Super CD-ROM 15 (1997)(EMAP Images)(GB)[!][issue 1997-10].iso / CUCD / Graphics / Ghostscript / source / gdevbbox.c < prev    next >
C/C++ Source or Header  |  1997-06-20  |  23KB  |  749 lines

  1. /* Copyright (C) 1996, 1997 Aladdin Enterprises.  All rights reserved.
  2.   
  3.   This file is part of Aladdin Ghostscript.
  4.   
  5.   Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND.  No author
  6.   or distributor accepts any responsibility for the consequences of using it,
  7.   or for whether it serves any particular purpose or works at all, unless he
  8.   or she says so in writing.  Refer to the Aladdin Ghostscript Free Public
  9.   License (the "License") for full details.
  10.   
  11.   Every copy of Aladdin Ghostscript must include a copy of the License,
  12.   normally in a plain ASCII text file named PUBLIC.  The License grants you
  13.   the right to copy, modify and redistribute Aladdin Ghostscript, but only
  14.   under certain conditions described in the License.  Among other things, the
  15.   License requires that the copyright notice and this notice be preserved on
  16.   all copies.
  17. */
  18.  
  19. /* gdevbbox.c */
  20. /* Device for tracking bounding box */
  21. #include "math_.h"
  22. #include "memory_.h"
  23. #include "gx.h"
  24. #include "gserrors.h"
  25. #include "gsparam.h"
  26. #include "gxdevice.h"
  27. #include "gsdevice.h"        /* requires gsmatrix.h */
  28. #include "gdevbbox.h"
  29. #include "gxistate.h"
  30. #include "gxpaint.h"
  31. #include "gxpath.h"
  32. #include "gxcpath.h"
  33.  
  34. /* Define TEST to create an output_page procedure for testing. */
  35. /*#define TEST*/
  36.  
  37. /* GC descriptor */
  38. public_st_device_bbox();
  39.  
  40. /* Device procedures */
  41. private dev_proc_open_device(bbox_open_device);
  42. private dev_proc_close_device(gx_forward_close_device);    /* see below */
  43. #ifdef TEST
  44. private dev_proc_output_page(bbox_output_page);
  45. #else
  46. #define bbox_output_page gx_forward_output_page
  47. #endif
  48. private dev_proc_fill_rectangle(bbox_fill_rectangle);
  49. private dev_proc_copy_mono(bbox_copy_mono);
  50. private dev_proc_copy_color(bbox_copy_color);
  51. private dev_proc_draw_line(bbox_draw_line);
  52. private dev_proc_get_params(bbox_get_params);
  53. private dev_proc_put_params(bbox_put_params);
  54. private dev_proc_copy_alpha(bbox_copy_alpha);
  55. private dev_proc_fill_path(bbox_fill_path);
  56. private dev_proc_stroke_path(bbox_stroke_path);
  57. private dev_proc_fill_mask(bbox_fill_mask);
  58. private dev_proc_fill_trapezoid(bbox_fill_trapezoid);
  59. private dev_proc_fill_parallelogram(bbox_fill_parallelogram);
  60. private dev_proc_fill_triangle(bbox_fill_triangle);
  61. private dev_proc_draw_thin_line(bbox_draw_thin_line);
  62. private dev_proc_begin_image(bbox_begin_image);
  63. private dev_proc_image_data(bbox_image_data);
  64. private dev_proc_end_image(bbox_end_image);
  65. private dev_proc_strip_tile_rectangle(bbox_strip_tile_rectangle);
  66. private dev_proc_strip_copy_rop(bbox_strip_copy_rop);
  67.  
  68. /* The device prototype */
  69. #ifndef TEST
  70. private const
  71. #endif
  72. /* We initialize with a very high resolution, to prevent limitchecks */
  73. /* if the resolution is changed to any reasonable value. */
  74. /* We initialize the width and height to "infinite" values, */
  75. /* leaving a little room for stroke widths, rounding, etc. */
  76. #define max_coord (min(max_int, fixed2int(max_fixed)) - 1000)
  77. #define max_resolution 4000
  78. gx_device_bbox far_data gs_bbox_device = {
  79.     std_device_std_body(gx_device_bbox, 0, "bbox",
  80.                 max_coord, max_coord,
  81.                 max_resolution, max_resolution),
  82.     {    bbox_open_device,
  83.         NULL,            /* get_initial_matrix */
  84.         NULL,            /* sync_output */
  85.         bbox_output_page,
  86.         gx_forward_close_device,
  87.         NULL,            /* map_rgb_color */
  88.         NULL,            /* map_color_rgb */
  89.         bbox_fill_rectangle,
  90.         NULL,            /* tile_rectangle */
  91.         bbox_copy_mono,
  92.         bbox_copy_color,
  93.         bbox_draw_line,
  94.         NULL,            /* get_bits */
  95.         bbox_get_params,
  96.         bbox_put_params,
  97.         NULL,            /* map_cmyk_color */
  98.         NULL,            /* get_xfont_procs */
  99.         NULL,            /* get_xfont_device */
  100.         NULL,            /* map_rgb_alpha_color */
  101.         gx_page_device_get_page_device,
  102.         NULL,            /* get_alpha_bits */
  103.         bbox_copy_alpha,
  104.         NULL,            /* get_band */
  105.         NULL,            /* copy_rop */
  106.         bbox_fill_path,
  107.         bbox_stroke_path,
  108.         bbox_fill_mask,
  109.         bbox_fill_trapezoid,
  110.         bbox_fill_parallelogram,
  111.         bbox_fill_triangle,
  112.         bbox_draw_thin_line,
  113.         bbox_begin_image,
  114.         bbox_image_data,
  115.         bbox_end_image,
  116.         bbox_strip_tile_rectangle,
  117.         bbox_strip_copy_rop
  118.     },
  119.     0                /* target */
  120. };
  121. #undef max_coord
  122. #undef max_resolution
  123.  
  124. /* Copy device parameters back from the target. */
  125. private void
  126. bbox_copy_params(gx_device_bbox *bdev)
  127. {    gx_device *tdev = bdev->target;
  128.     if ( tdev != 0 )
  129.       { /* This is kind of scatter-shot.... */
  130. #define copy_param(p) bdev->p = tdev->p
  131. #define copy_array_param(p) memcpy(bdev->p, tdev->p, sizeof(bdev->p))
  132.         copy_param(width);
  133.         copy_param(height);
  134.         copy_array_param(MediaSize);
  135.         copy_array_param(ImagingBBox);
  136.         copy_param(ImagingBBox_set);
  137.         copy_array_param(HWResolution);
  138.         copy_array_param(MarginsHWResolution);
  139.         copy_array_param(Margins);
  140.         copy_array_param(HWMargins);
  141.         copy_param(color_info);
  142. #undef copy_param
  143. #undef copy_array_param
  144.       }
  145.     if ( dev_proc(bdev, map_rgb_color) != 0 )
  146.       bdev->white =
  147.         (*dev_proc(bdev, map_rgb_color))
  148.           ((gx_device *)bdev, gx_max_color_value, gx_max_color_value,
  149.            gx_max_color_value);
  150. }
  151.  
  152. #define bdev ((gx_device_bbox *)dev)
  153.  
  154. #define gx_dc_is_white(pdevc, bdev)\
  155.   (gx_dc_is_pure(pdevc) && gx_dc_pure_color(pdevc) == (bdev)->white)
  156.  
  157. /* Note that some of the "forward" procedures don't exist. */
  158. /* We open-code all but this one below. */
  159. private int
  160. gx_forward_close_device(gx_device *dev)
  161. {    gx_device *tdev = bdev->target;
  162.     return (tdev == 0 ? 0 : (*dev_proc(tdev, close_device))(tdev));
  163. }
  164.  
  165. /* Bounding box utilities */
  166.  
  167. private void near
  168. bbox_initialize(gs_fixed_rect *pr)
  169. {    pr->p.x = pr->p.y = max_fixed;
  170.     pr->q.x = pr->q.y = min_fixed;
  171. }
  172.  
  173. private void near
  174. bbox_add_rect(gs_fixed_rect *pr, fixed x0, fixed y0, fixed x1, fixed y1)
  175. {    if ( x0 < pr->p.x )
  176.       pr->p.x = x0;
  177.     if ( y0 < pr->p.y )
  178.       pr->p.y = y0;
  179.     if ( x1 > pr->q.x )
  180.       pr->q.x = x1;
  181.     if ( y1 > pr->q.y )
  182.       pr->q.y = y1;
  183. }
  184. private void near
  185. bbox_add_point(gs_fixed_rect *pr, fixed x, fixed y)
  186. {    bbox_add_rect(pr, x, y, x, y);
  187. }
  188. private void near
  189. bbox_add_int_rect(gs_fixed_rect *pr, int x0, int y0, int x1, int y1)
  190. {    bbox_add_rect(pr, int2fixed(x0), int2fixed(y0), int2fixed(x1),
  191.               int2fixed(y1));
  192. }
  193.  
  194. #define rect_is_page(dev, x, y, w, h)\
  195.   (x <= 0 && y <= 0 && w >= x + dev->width && h >= y + dev->height)
  196.  
  197. /* ---------------- Open/close/page ---------------- */
  198.  
  199. /* Initialize a bounding box device. */
  200. void
  201. gx_device_bbox_init(gx_device_bbox *dev, gx_device *target)
  202. {    *dev = gs_bbox_device;
  203.     gx_device_forward_fill_in_procs((gx_device_forward *)dev);
  204.     bdev->target = target;
  205.     bbox_copy_params(dev);
  206. }
  207.  
  208. /* Read back the bounding box in 1/72" units. */
  209. void
  210. gx_device_bbox_bbox(gx_device_bbox *dev, gs_rect *pbbox)
  211. {    gs_matrix mat;
  212.     gs_rect dbox;
  213.  
  214.     gs_deviceinitialmatrix((gx_device *)dev, &mat);
  215.     dbox.p.x = fixed2float(bdev->bbox.p.x);
  216.     dbox.p.y = fixed2float(bdev->bbox.p.y);
  217.     dbox.q.x = fixed2float(bdev->bbox.q.x);
  218.     dbox.q.y = fixed2float(bdev->bbox.q.y);
  219.     gs_bbox_transform_inverse(&dbox, &mat, pbbox);
  220. }
  221.  
  222.  
  223. private int
  224. bbox_open_device(gx_device *dev)
  225. {    bbox_initialize(&bdev->bbox);
  226. #ifdef TEST
  227.     gx_device_forward_fill_in_procs((gx_device_forward *)dev);
  228. #endif
  229.     /* gx_forward_open_device doesn't exist */
  230.     { gx_device *tdev = bdev->target;
  231.       int code = (tdev == 0 ? 0 : (*dev_proc(tdev, open_device))(tdev));
  232.       bbox_copy_params(bdev);
  233.       return code;
  234.     }
  235. }
  236.  
  237. #ifdef TEST
  238. private int 
  239. bbox_output_page(gx_device *dev, int num_copies, int flush)
  240. {    gs_rect bbox;
  241.  
  242.     /* Print the page bounding box. */
  243.     gx_device_bbox_bbox((gx_device_bbox *)dev, &bbox);
  244.     dprintf2("[gdevbbox] lower left  = %f %f\n", bbox.p.x, bbox.p.y);
  245.     dprintf2("[gdevbbox] upper right = %f %f\n", bbox.q.x, bbox.q.y);
  246.     return gx_forward_output_page(dev, num_copies, flush);
  247. }
  248. #endif
  249.  
  250. /* ---------------- Low-level drawing ---------------- */
  251.  
  252. private int
  253. bbox_fill_rectangle(gx_device *dev, int x, int y, int w, int h,
  254.   gx_color_index color)
  255. {    /* Check for erasing the entire page. */
  256.     if ( rect_is_page(dev, x, y, w, h) )
  257.       bbox_initialize(&bdev->bbox);
  258.     else if ( color != bdev->white )
  259.       bbox_add_int_rect(&bdev->bbox, x, y, x + w, y + h);
  260.     /* gx_forward_fill_rectangle doesn't exist */
  261.     { gx_device *tdev = bdev->target;
  262.       return (tdev == 0 ? 0 :
  263.           (*dev_proc(tdev, fill_rectangle))(tdev, x, y, w, h, color));
  264.     }
  265. }
  266.  
  267. private int
  268. bbox_copy_mono(gx_device *dev, const byte *data,
  269.   int dx, int raster, gx_bitmap_id id, int x, int y, int w, int h,
  270.   gx_color_index zero, gx_color_index one)
  271. {    if ( (one != gx_no_color_index && one != bdev->white) ||
  272.          (zero != gx_no_color_index && zero != bdev->white)
  273.        )
  274.       bbox_add_int_rect(&bdev->bbox, x, y, x + w, y + h);
  275.     /* gx_forward_copy_mono doesn't exist */
  276.     { gx_device *tdev = bdev->target;
  277.       return (tdev == 0 ? 0 :
  278.           (*dev_proc(tdev, copy_mono))(tdev, data, dx, raster, id,
  279.                            x, y, w, h, zero, one));
  280.     }
  281. }
  282.  
  283. private int
  284. bbox_copy_color(gx_device *dev, const byte *data,
  285.   int dx, int raster, gx_bitmap_id id, int x, int y, int w, int h)
  286. {    bbox_add_int_rect(&bdev->bbox, x, y, x + w, y + h);
  287.     /* gx_forward_copy_color doesn't exist */
  288.     { gx_device *tdev = bdev->target;
  289.       return (tdev == 0 ? 0 :
  290.           (*dev_proc(tdev, copy_color))(tdev, data, dx, raster, id,
  291.                            x, y, w, h));
  292.     }
  293. }
  294.  
  295. private int
  296. bbox_draw_line(gx_device *dev,
  297.   int x0, int y0, int x1, int y1, gx_color_index color)
  298. {    int xp, yp, xq, yq;
  299.     if ( x0 < x1 )
  300.       xp = x0, xq = x1 + 1;
  301.     else
  302.       xp = x1, xq = x0 + 1;
  303.     if ( y0 < y1 )
  304.       yp = y0, yq = y1 + 1;
  305.     else
  306.       yp = y1, yq = y0 + 1;
  307.     if ( color != bdev->white )
  308.       bbox_add_int_rect(&bdev->bbox, xp, yp, xq, yq);
  309.     /* gx_forward_draw_line doesn't exist */
  310.     { gx_device *tdev = bdev->target;
  311.       return (tdev == 0 ? 0 :
  312.           (*dev_proc(tdev, draw_line))(tdev, x0, y0, x1, y1, color));
  313.     }
  314. }
  315.  
  316. private int
  317. bbox_copy_alpha(gx_device *dev, const byte *data, int data_x,
  318.   int raster, gx_bitmap_id id, int x, int y, int w, int h,
  319.   gx_color_index color, int depth)
  320. {    bbox_add_int_rect(&bdev->bbox, x, y, x + w, y + h);
  321.     /* gx_forward_copy_alpha doesn't exist */
  322.     { gx_device *tdev = bdev->target;
  323.       return (tdev == 0 ? 0 :
  324.           (*dev_proc(tdev, copy_alpha))(tdev, data, data_x, raster, id,
  325.                         x, y, w, h, color, depth));
  326.     }
  327. }
  328.  
  329. private int
  330. bbox_strip_tile_rectangle(gx_device *dev, const gx_strip_bitmap *tiles,
  331.   int x, int y, int w, int h, gx_color_index color0, gx_color_index color1,
  332.   int px, int py)
  333. {    if ( rect_is_page(dev, x, y, w, h) )
  334.       bbox_initialize(&bdev->bbox);
  335.     else
  336.       bbox_add_int_rect(&bdev->bbox, x, y, x + w, y + h);
  337.     /* Skip the call if there is no target. */
  338.     { gx_device *tdev = bdev->target;
  339.       return (tdev == 0 ? 0 :
  340.           (*dev_proc(tdev, strip_tile_rectangle))(tdev, tiles, x, y,
  341.                         w, h, color0, color1, px, py));
  342.     }
  343. }
  344.  
  345. private int
  346. bbox_strip_copy_rop(gx_device *dev,
  347.   const byte *sdata, int sourcex, uint sraster, gx_bitmap_id id,
  348.   const gx_color_index *scolors,
  349.   const gx_strip_bitmap *textures, const gx_color_index *tcolors,
  350.   int x, int y, int w, int h,
  351.   int phase_x, int phase_y, gs_logical_operation_t lop)
  352. {    bbox_add_int_rect(&bdev->bbox, x, y, x + w, y + h);
  353.     /* gx_forward_strip_copy_rop doesn't exist */
  354.     { gx_device *tdev = bdev->target;
  355.       return (tdev == 0 ? 0 :
  356.           (*dev_proc(tdev, strip_copy_rop))(tdev,
  357.                           sdata, sourcex, sraster, id,
  358.                           scolors, textures, tcolors,
  359.                           x, y, w, h,
  360.                           phase_x, phase_y, lop));
  361.     }
  362. }
  363.  
  364. /* ---------------- Parameters ---------------- */
  365.  
  366. /* We implement get_params to provide a way to read out the bounding box. */
  367. private int
  368. bbox_get_params(gx_device *dev, gs_param_list *plist)
  369. {    int code = gx_forward_get_params(dev, plist);
  370.     gs_param_float_array bba;
  371.     float bbox[4];
  372.  
  373.     if ( code < 0 )
  374.       return code;
  375.     bbox[0] = fixed2float(bdev->bbox.p.x);
  376.     bbox[1] = fixed2float(bdev->bbox.p.y);
  377.     bbox[2] = fixed2float(bdev->bbox.q.x);
  378.     bbox[3] = fixed2float(bdev->bbox.q.y);
  379.     bba.data = bbox, bba.size = 4, bba.persistent = false;
  380.     return param_write_float_array(plist, "PageBoundingBox", &bba);
  381. }
  382.  
  383. /* We implement put_params to ensure that we keep the important */
  384. /* device parameters up to date, and to prevent an /undefined error */
  385. /* from PageBoundingBox. */
  386. private int
  387. bbox_put_params(gx_device *dev, gs_param_list *plist)
  388. {    int code;
  389.     int ecode = 0;
  390.     gs_param_name param_name;
  391.     gs_param_float_array bba;
  392.  
  393.     code = param_read_float_array(plist, (param_name = "PageBoundingBox"),
  394.                       &bba);
  395.     switch ( code )
  396.       {
  397.       case 0:
  398.         if ( bba.size != 4 )
  399.           { ecode = gs_note_error(gs_error_rangecheck);
  400.             goto e;
  401.           }
  402.         break;
  403.       default:
  404.         ecode = code;
  405. e:        param_signal_error(plist, param_name, ecode);
  406.       case 1:
  407.         bba.data = 0;
  408.       }
  409.  
  410.     code = gx_forward_put_params(dev, plist);
  411.     if ( ecode < 0 )
  412.       code = ecode;
  413.     if ( code >= 0 && bba.data != 0 )
  414.       { bdev->bbox.p.x = float2fixed(bba.data[0]);
  415.         bdev->bbox.p.y = float2fixed(bba.data[1]);
  416.         bdev->bbox.q.x = float2fixed(bba.data[2]);
  417.         bdev->bbox.q.y = float2fixed(bba.data[3]);
  418.       }
  419.     bbox_copy_params(bdev);
  420.     return code;
  421. }
  422.  
  423. /* ---------------- Polygon drawing ---------------- */
  424.  
  425. private fixed
  426. edge_x_at_y(const gs_fixed_edge *edge, fixed y)
  427. {    return fixed_mult_quo(edge->end.x - edge->start.x,
  428.                   y - edge->start.y,
  429.                   edge->end.y - edge->start.y) + edge->start.x;
  430. }
  431. private int
  432. bbox_fill_trapezoid(gx_device *dev,
  433.   const gs_fixed_edge *left, const gs_fixed_edge *right,
  434.   fixed ybot, fixed ytop, bool swap_axes,
  435.   const gx_device_color *pdevc, gs_logical_operation_t lop)
  436. {    if ( !gx_dc_is_white(pdevc, bdev) )
  437.       { fixed x0l =
  438.           (left->start.y == ybot ? left->start.x :
  439.            edge_x_at_y(left, ybot));
  440.         fixed x1l =
  441.           (left->end.y == ytop ? left->end.x :
  442.            edge_x_at_y(left, ytop));
  443.         fixed x0r =
  444.           (right->start.y == ybot ? right->start.x :
  445.            edge_x_at_y(right, ybot));
  446.         fixed x1r =
  447.           (right->end.y == ytop ? right->end.x :
  448.            edge_x_at_y(right, ytop));
  449.         fixed xminl = min(x0l, x1l), xmaxl = max(x0l, x1l);
  450.         fixed xminr = min(x0r, x1r), xmaxr = max(x0r, x1r);
  451.         fixed x0 = min(xminl, xminr), x1 = max(xmaxl, xmaxr);
  452.  
  453.         if ( swap_axes )
  454.           bbox_add_rect(&bdev->bbox, ybot, x0, ytop, x1);
  455.         else
  456.           bbox_add_rect(&bdev->bbox, x0, ybot, x1, ytop);
  457.       }
  458.     /* Skip the call if there is no target. */
  459.     { gx_device *tdev = bdev->target;
  460.       return (tdev == 0 ? 0 :
  461.           (*dev_proc(tdev, fill_trapezoid))
  462.            (tdev, left, right, ybot, ytop, swap_axes, pdevc, lop));
  463.     }
  464. }
  465.  
  466. private int
  467. bbox_fill_parallelogram(gx_device *dev,
  468.   fixed px, fixed py, fixed ax, fixed ay, fixed bx, fixed by,
  469.   const gx_device_color *pdevc, gs_logical_operation_t lop)
  470. {    if ( !gx_dc_is_white(pdevc, bdev) )
  471.       { fixed pax = px + ax, pay = py + ay;
  472.         bbox_add_rect(&bdev->bbox, px, py, px + bx, py + by);
  473.         bbox_add_rect(&bdev->bbox, pax, pay, pax + bx, pay + by);
  474.       }
  475.     /* Skip the call if there is no target. */
  476.     { gx_device *tdev = bdev->target;
  477.       return (tdev == 0 ? 0 :
  478.           (*dev_proc(tdev, fill_parallelogram))(tdev, px, py, ax, ay,
  479.                             bx, by, pdevc, lop));
  480.     }
  481. }
  482.  
  483. private int
  484. bbox_fill_triangle(gx_device *dev,
  485.   fixed px, fixed py, fixed ax, fixed ay, fixed bx, fixed by,
  486.   const gx_device_color *pdevc, gs_logical_operation_t lop)
  487. {    if ( !gx_dc_is_white(pdevc, bdev) )
  488.       { bbox_add_rect(&bdev->bbox, px, py, px + bx, py + by);
  489.         bbox_add_point(&bdev->bbox, px + ax, py + ay);
  490.       }
  491.     /* Skip the call if there is no target. */
  492.     { gx_device *tdev = bdev->target;
  493.       return (tdev == 0 ? 0 :
  494.           (*dev_proc(tdev, fill_triangle))(tdev, px, py, ax, ay,
  495.                            bx, by, pdevc, lop));
  496.     }
  497. }
  498.  
  499. private int
  500. bbox_draw_thin_line(gx_device *dev,
  501.   fixed fx0, fixed fy0, fixed fx1, fixed fy1,
  502.   const gx_device_color *pdevc, gs_logical_operation_t lop)
  503. {    if ( !gx_dc_is_white(pdevc, bdev) )
  504.       bbox_add_rect(&bdev->bbox, fx0, fy0, fx1, fy1);
  505.     /* Skip the call if there is no target. */
  506.     { gx_device *tdev = bdev->target;
  507.       return (tdev == 0 ? 0 :
  508.           (*dev_proc(tdev, draw_thin_line))(tdev, fx0, fy0, fx1, fy0,
  509.                             pdevc, lop));
  510.     }
  511. }
  512.  
  513. /* ---------------- High-level drawing ---------------- */
  514.  
  515. #define adjust_box(pbox, adj)\
  516.   ((pbox)->p.x -= (adj).x, (pbox)->p.y -= (adj).y,\
  517.    (pbox)->q.x += (adj).x, (pbox)->q.y += (adj).y)
  518.  
  519. private int
  520. bbox_fill_path(gx_device *dev, const gs_imager_state *pis, gx_path *ppath,
  521.   const gx_fill_params *params, const gx_device_color *pdevc,
  522.   const gx_clip_path *pcpath)
  523. {    gx_device *tdev = bdev->target;
  524.  
  525.     if ( !gx_dc_is_white(pdevc, bdev) )
  526.       { gs_fixed_rect ibox;
  527.         gs_fixed_point adjust;
  528.  
  529.         if ( gx_path_bbox(ppath, &ibox) < 0 )
  530.           return 0;
  531.         adjust = params->adjust;
  532.         if ( params->fill_zero_width )
  533.           gx_adjust_if_empty(&ibox, &adjust);
  534.         adjust_box(&ibox, adjust);
  535.         if ( pcpath != NULL &&
  536.          !gx_cpath_includes_rectangle(pcpath, ibox.p.x, ibox.p.y,
  537.                           ibox.q.x, ibox.q.y)
  538.            )
  539.           { /* Let the target do the drawing, but break down the */
  540.         /* fill path into pieces for computing the bounding box. */
  541.         bdev->target = NULL;
  542.         gx_default_fill_path(dev, pis, ppath, params, pdevc, pcpath);
  543.         bdev->target = tdev;
  544.           }
  545.         else
  546.           { /* Just use the path bounding box. */
  547.         bbox_add_rect(&bdev->bbox, ibox.p.x, ibox.p.y, ibox.q.x,
  548.                   ibox.q.y);
  549.           }
  550.       }
  551.     /* Skip the call if there is no target. */
  552.     return (tdev == 0 ? 0 :
  553.         (*dev_proc(tdev, fill_path))(tdev, pis, ppath, params, pdevc,
  554.                          pcpath));
  555. }
  556.  
  557. private int
  558. bbox_stroke_path(gx_device *dev, const gs_imager_state *pis, gx_path *ppath,
  559.   const gx_stroke_params *params,
  560.   const gx_drawing_color *pdevc, const gx_clip_path *pcpath)
  561. {    gx_device *tdev = bdev->target;
  562.  
  563.     if ( !gx_dc_is_white(pdevc, bdev) )
  564.       { gs_fixed_rect ibox;
  565.         gs_fixed_point expand;
  566.  
  567.         if ( gx_path_bbox(ppath, &ibox) < 0 )
  568.           return 0;
  569.         if ( gx_stroke_path_expansion(pis, ppath, &expand) < 0 )
  570.           ibox.p.x = ibox.p.y = min_fixed, ibox.q.x = ibox.q.y = max_fixed;
  571.         else
  572.           adjust_box(&ibox, expand);
  573.         if ( pcpath != NULL &&
  574.          !gx_cpath_includes_rectangle(pcpath, ibox.p.x, ibox.p.y,
  575.                           ibox.q.x, ibox.q.y)
  576.            )
  577.           { /* Let the target do the drawing, but break down the */
  578.         /* fill path into pieces for computing the bounding box. */
  579.         bdev->target = NULL;
  580.         gx_default_stroke_path(dev, pis, ppath, params, pdevc, pcpath);
  581.         bdev->target = tdev;
  582.           }
  583.         else
  584.           { /* Just use the path bounding box. */
  585.         bbox_add_rect(&bdev->bbox, ibox.p.x, ibox.p.y, ibox.q.x,
  586.                   ibox.q.y);
  587.           }
  588.       }
  589.     /* Skip the call if there is no target. */
  590.     return (tdev == 0 ? 0 :
  591.         (*dev_proc(tdev, stroke_path))(tdev, pis, ppath, params,
  592.                            pdevc, pcpath));
  593. }
  594.  
  595. private int
  596. bbox_fill_mask(gx_device *dev,
  597.   const byte *data, int dx, int raster, gx_bitmap_id id,
  598.   int x, int y, int w, int h,
  599.   const gx_drawing_color *pdcolor, int depth,
  600.   gs_logical_operation_t lop, const gx_clip_path *pcpath)
  601. {    gx_device *tdev = bdev->target;
  602.  
  603.     if ( pcpath != NULL &&
  604.          !gx_cpath_includes_rectangle(pcpath, int2fixed(x), int2fixed(y),
  605.                       int2fixed(x + w),
  606.                       int2fixed(y + h))
  607.        )
  608.       { /* Let the target do the drawing, but break down the */
  609.         /* image into pieces for computing the bounding box. */
  610.         bdev->target = NULL;
  611.         gx_default_fill_mask(dev, data, dx, raster, id, x, y, w, h,
  612.                  pdcolor, depth, lop, pcpath);
  613.         bdev->target = tdev;
  614.       }
  615.     else
  616.       { /* Just use the mask bounding box. */
  617.         bbox_add_int_rect(&bdev->bbox, x, y, x + w, y + h);
  618.       }
  619.     /* Skip the call if there is no target. */
  620.     return (tdev == 0 ? 0 :
  621.         (*dev_proc(tdev, fill_mask))(tdev, data, dx, raster, id, x, y,
  622.                      w, h, pdcolor, depth, lop, pcpath));
  623. }
  624.  
  625. /* ------ Bitmap imaging ------ */
  626.  
  627. typedef struct bbox_image_enum_s {
  628.     gs_memory_t *memory;
  629.     gs_matrix matrix;    /* map from image space to device space */
  630.     const gx_clip_path *pcpath;
  631.     void *target_info;
  632.     int x0, x1;
  633.     int y, height;
  634. } bbox_image_enum;
  635. gs_private_st_ptrs2(st_bbox_image_enum, bbox_image_enum, "bbox_image_enum",
  636.   bbox_image_enum_enum_ptrs, bbox_image_enum_reloc_ptrs, pcpath, target_info);
  637.  
  638. private int
  639. bbox_begin_image(gx_device *dev,
  640.   const gs_imager_state *pis, const gs_image_t *pim,
  641.   gs_image_format_t format, const gs_int_rect *prect,
  642.   const gx_drawing_color *pdcolor, const gx_clip_path *pcpath,
  643.   gs_memory_t *memory, void **pinfo)
  644. {    int code;
  645.     gs_matrix mat;
  646.     bbox_image_enum *pbe;
  647.  
  648.     if ( (code = gs_matrix_invert(&pim->ImageMatrix, &mat)) < 0 ||
  649.          (code = gs_matrix_multiply(&mat, &ctm_only(pis), &mat)) < 0
  650.        )
  651.       return_error(gs_error_rangecheck);
  652.     pbe = gs_alloc_struct(memory, bbox_image_enum, &st_bbox_image_enum,
  653.                   "bbox_begin_image");
  654.     if ( pbe == 0 )
  655.       return_error(gs_error_VMerror);
  656.     pbe->memory = memory;
  657.     pbe->matrix = mat;
  658.     pbe->pcpath = pcpath;
  659.     pbe->target_info = 0;        /* in case no target */
  660.     if ( prect )
  661.       { pbe->x0 = prect->p.x, pbe->x1 = prect->q.x;
  662.         pbe->y = prect->p.y, pbe->height = prect->q.y - prect->p.y;
  663.       }
  664.     else
  665.       { pbe->x0 = 0, pbe->x1 = pim->Width;
  666.         pbe->y = 0, pbe->height = pim->Height;
  667.       }
  668.     *pinfo = pbe;
  669.     /* Skip the call if there is no target. */
  670.     { gx_device *tdev = bdev->target;
  671.       return (tdev == 0 ? 0 :
  672.           (*dev_proc(tdev, begin_image))(tdev, pis, pim, format, prect,
  673.                          pdcolor, pcpath, memory,
  674.                          &pbe->target_info));
  675.     }
  676. }
  677.  
  678. private int
  679. bbox_image_data(gx_device *dev,
  680.   void *info, const byte **planes, int data_x, uint raster, int height)
  681. {    gx_device *tdev = bdev->target;
  682.     bbox_image_enum *pbe = info;
  683.     const gx_clip_path *pcpath = pbe->pcpath;
  684.     gs_rect sbox, dbox;
  685.     gs_point corners[4];
  686.     gs_fixed_rect ibox;
  687.  
  688.     sbox.p.x = pbe->x0;
  689.     sbox.p.y = pbe->y;
  690.     sbox.q.x = pbe->x1;
  691.     sbox.q.y = pbe->y += height;
  692.     gs_bbox_transform_only(&sbox, &pbe->matrix, corners);
  693.     gs_points_bbox(corners, &dbox);
  694.     ibox.p.x = float2fixed(dbox.p.x);
  695.     ibox.p.y = float2fixed(dbox.p.y);
  696.     ibox.q.x = float2fixed(dbox.q.x);
  697.     ibox.q.y = float2fixed(dbox.q.y);
  698.     if ( pcpath != NULL &&
  699.          !gx_cpath_includes_rectangle(pcpath, ibox.p.x, ibox.p.y,
  700.                       ibox.q.x, ibox.q.y)
  701.        )
  702.       { /* Let the target do the drawing, but drive two triangles */
  703.         /* through the clipping path to get an accurate bounding box. */
  704.         gx_device_clip cdev;
  705.         gx_drawing_color devc;
  706.         fixed x0 = float2fixed(corners[0].x),
  707.           y0 = float2fixed(corners[0].y);
  708.         fixed bx2 = float2fixed(corners[2].x) - x0,
  709.           by2 = float2fixed(corners[2].y) - y0;
  710.  
  711.         gx_make_clip_path_device(&cdev, pcpath);
  712.         cdev.target = dev;
  713.         (*dev_proc(&cdev, open_device))((gx_device *)&cdev);
  714.         color_set_pure(&devc, 0);        /* any color will do */
  715.         bdev->target = NULL;
  716.         gx_default_fill_triangle((gx_device *)&cdev, x0, y0,
  717.                      float2fixed(corners[1].x) - x0,
  718.                      float2fixed(corners[1].y) - y0,
  719.                      bx2, by2, &devc, lop_default);
  720.         gx_default_fill_triangle((gx_device *)&cdev, x0, y0,
  721.                      float2fixed(corners[3].x) - x0,
  722.                      float2fixed(corners[3].y) - y0,
  723.                      bx2, by2, &devc, lop_default);
  724.         bdev->target = tdev;
  725.       }
  726.     else
  727.       { /* Just use the bounding box. */
  728.         bbox_add_rect(&bdev->bbox, ibox.p.x, ibox.p.y, ibox.q.x, ibox.q.y);
  729.       }
  730.     /* Skip the call if there is no target. */
  731.     return (tdev == 0 ? pbe->y >= pbe->height :
  732.         (*dev_proc(tdev, image_data))(tdev, pbe->target_info, planes,
  733.                           data_x, raster, height));
  734. }
  735.  
  736. private int
  737. bbox_end_image(gx_device *dev, void *info, bool draw_last)
  738. {    bbox_image_enum *pbe = info;
  739.     void *target_info = pbe->target_info;
  740.     /* Skip the call if there is no target. */
  741.     gx_device *tdev = bdev->target;
  742.     int code =
  743.       (tdev == 0 ? 0 :
  744.        (*dev_proc(tdev, end_image))(tdev, target_info, draw_last));
  745.  
  746.     gs_free_object(pbe->memory, pbe, "bbox_end_image");
  747.     return code;
  748. }
  749.